package fr.ippon.spark.metrics;
import com.fasterxml.jackson.core.JsonFactory;
import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;
import org.apache.spark.storage.StorageLevel;
import org.apache.spark.streaming.receiver.Receiver;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.net.*;
import java.util.HashMap;
/**
* A Spark Streaming receiver which reads metric values sent by a Metrics SparkReporter.
*
*/
public class MetricsReceiver extends Receiver<HashMap<String, Object>> {
private static final Logger LOGGER = LoggerFactory.getLogger(MetricsReceiver.class);
private ServerSocket sparkSocket;
private Socket metricsSocket;
private BufferedReader reader;
private int port;
private StorageLevel storageLevel;
public MetricsReceiver(int port) {
super(StorageLevel.MEMORY_ONLY());
this.port = port;
this.storageLevel = StorageLevel.MEMORY_ONLY();
}
public MetricsReceiver(int port, StorageLevel storageLevel) {
super(storageLevel);
this.port = port;
this.storageLevel = storageLevel;
sparkSocket = null;
metricsSocket = null;
reader = null;
}
@Override
public StorageLevel storageLevel() {
return storageLevel;
}
/**
* This method is called by the system when the receiver is started.
*/
@Override
public void onStart() {
new Thread() {
@Override
public void run() {
receive();
}
}.start();
}
/**
* This method is called by the system when the receiver is stopped.
*/
@Override
public void onStop() {
try {
close();
} catch (IOException ioe) {
LOGGER.warn("Could not stop the receiver properly : ", ioe.getClass().getCanonicalName());
}
}
private void receive() {
try {
connect();
ObjectMapper mapper = new ObjectMapper(new JsonFactory());
String input;
HashMap<String, Object> map;
while (!isStopped() && (input = reader.readLine()) != null) {
map = mapper.readValue(input, new TypeReference<HashMap<String, Object>>() {});
store(map);
}
close();
} catch (IOException ioe) {
LOGGER.warn("Unable to receive from Metrics : ", ioe.getClass().getCanonicalName());
receive();
}
}
private void connect() throws IOException {
if (sparkSocket != null || metricsSocket != null || reader != null) {
close();
}
sparkSocket = new ServerSocket(port);
metricsSocket = sparkSocket.accept();
reader = new BufferedReader(new InputStreamReader(metricsSocket.getInputStream()));
}
private void close() throws IOException {
reader.close();
metricsSocket.close();
sparkSocket.close();
reader = null;
metricsSocket = null;
sparkSocket = null;
}
}